#pragma once

// MySQL版

#include <iostream>
#include <string>
#include <unordered_map>
#include <vector>
#include <cassert>
#include <fstream>

#include <mysql/mysql.h>

#include "../comm/log.hpp"
//#include "include/mysql.h"
#include "../comm/util.hpp"


// 根据题目list文件，加载所有的题目信息到内存中
// model: 只要用来和数据进行交互，对外提供访问数据的接口

namespace ns_model
{
    using namespace std;
    using namespace ns_log;
    using namespace ns_util;

    struct Question
    {
        std::string number;     // 题目编号，唯一
        std::string title;      // 题目标题
        std::string star;       // 题目星级（难度）
        std::string desc;       // 题目描述
        std::string header;     // 题目预设给用户在线编辑器的代码
        std::string tail;       // 题目的测试用例，需要和header拼接，形成完整代码
        int cpu_limit;         // 时间要求（s） 
        int mem_limit;         // 内存限制（KB）
    };

    const std::string oj_questions = "oj_questions";    // 表名
    const std::string host = "127.0.0.1";
    const std::string user = "oj_client";
    const std::string password = "123456";
    const std::string db = "oj";
    const int port = 3306;

    class Model
    {
    public:
        Model()
        {}

        bool QueryMySql(const std::string &sql, vector<Question> *out)
        {
            // 创建mysql句柄
            MYSQL *my = mysql_init(nullptr);
            // unsigned int ssl_mode = SSL_MODE_DISABLED;
            // mysql_options(my, MYSQL_OPT_SSL_MODE, &ssl_mode);

            mysql_ssl_set(my, nullptr, nullptr, "ca.pem", nullptr,nullptr);

            // 连接数据库
            if(nullptr == mysql_real_connect(my, host.c_str(), \
                                user.c_str(),       \
                                password.c_str(),   \
                                db.c_str(),         \
                                port, nullptr, CLIENT_SSL))
            {
                LOG(FATAL) << mysql_error(my) <<  "   连接数据库失败！" << "\n";
                mysql_close(my);
                return false;
            }

            const char *ca = mysql_get_ssl_cipher(my);
            if(ca && *ca){
                LOG(INFO) << "ssl success :  " << ca << "\n";
            }else{
                LOG(FATAL) << "ssl failed" << "\n";
            }

            // 一定要设置连接的编码格式
            mysql_set_character_set(my, "utf8");

            LOG(INFO) << "连接数据库成功" << "\n";

            // 执行sql语句
            if(mysql_query(my, sql.c_str()))
            {
                LOG(WARNING) << sql << " execute error!" << "\n";
                mysql_close(my);
                return false;
            }

            // 提取结果
            MYSQL_RES *res = mysql_store_result(my);
            // 分析结果
            int rows = mysql_num_rows(res);     // 获得行数量
            int cols = mysql_num_fields(res);   // 获得列数量
            for (int i = 0; i < rows; i++)
            {
                MYSQL_ROW row = mysql_fetch_row(res);
                Question q;
                q.number = row[0];
                q.title = row[1];
                q.star = row[2];
                q.desc = row[3];
                q.header = row[4];
                q.tail = row[5];
                q.cpu_limit = atoi(row[6]);
                q.mem_limit = atoi(row[7]);
                out->push_back(q);
            }

            // 释放结果空间
            mysql_free_result(res);
            // 关闭mysql连接
            mysql_close(my);
            return true;
        }

        bool GetAllQuestions(std::vector<Question> *out)
        {
            std::string sql = "select * from ";
            sql += oj_questions;
            return QueryMySql(sql, out);
        }

        bool GetOneQuestion(const std::string &number, Question *q)
        {
            bool res = false;
            std::string sql = "select * from ";
            sql += oj_questions;
            sql += " where number=";
            sql += number;
            vector<Question> result;
            if(QueryMySql(sql, &result))
            {
                if(result.size() == 1)
                {
                    res = true; 
                    *q = result[0];
                }
            }
            return res;
        }

        ~Model()
        {}
    };
}